import { Alert, Platform, StyleSheet, } from 'react-native'; import ParallaxScrollView from '@/components/ParallaxScrollView'; import { ThemedText } from '@/components/ThemedText'; import { ThemedView } from '@/components/ThemedView'; import React, { useState, useCallback, useEffect, useRef } from 'react'; import AsyncStorage from '@react-native-async-storage/async-storage'; import { useFocusEffect } from '@react-navigation/native'; import { Link, router, useLocalSearchParams } from 'expo-router'; import MapView, { Marker, Polygon, Region } from 'react-native-maps'; import { Ionicons } from '@expo/vector-icons'; import { useColorScheme } from '@/hooks/useColorScheme.web'; import { Colors } from '@/constants/Colors'; interface AlertData { id: string; userId: string; createdAt: string; area: string; extendedArea: string; level: string; } interface PositionData { id: string; userId: string; createdAt: string; latitude: number; longitude: number; movingActivity: string; } interface NotificationData { id: string; alert: AlertData; position: PositionData; seen: boolean; createdAt: string; } interface PolygonCoordinates { latitude: number; longitude: number; } export default function NotificationIdScreen() { const [token, setToken] = useState(''); const [userId, setUserId] = useState(''); const [notification, setNotification] = useState(null); const [region, setRegion] = useState({ latitude: 44.49738301084014, longitude: 11.356121722966094, latitudeDelta: 0.03, longitudeDelta: 0.03, }); const [coordinates, setCoordinates] = useState({ latitude: 44.49738301084014, longitude: 11.356121722966094 }); const [polygon, setPolygon] = useState([]); const [extendedPolygon, setExtendedPolygon] = useState([]); const mapRef = useRef(null); const checkAuth = useCallback(async () => { const storedToken = Platform.OS === 'web' ? localStorage.getItem('token') : await AsyncStorage.getItem('token'); const storedUserId = Platform.OS === 'web' ? localStorage.getItem('userId') : await AsyncStorage.getItem('userId'); setToken(storedToken || ''); setUserId(storedUserId || ''); if (!storedToken || !storedUserId) { Alert.alert( 'Login required', 'You must log in to the system if you want to see notifications list', [ { text: 'Ok', onPress: () => router.push('/'), }, ] ); } }, []); const updateSeenStatus = async (id: string) => { const response = await fetch(`${process.env.EXPO_PUBLIC_API_URL}/graphql`, { method: 'POST', headers: { 'Authorization': `Bearer ${token}`, 'Content-Type': 'application/json', }, body: JSON.stringify({ query: ` mutation NotificationUpdate($input: NotificationUpdateInput!) { notificationUpdate(input: $input) { id seen } }`, variables: { input: { id, seen: true, }, }, }), }); const data = await response.json(); if (data.errors) { console.error(`On editing notification "${id}": ${JSON.stringify(data)}`); } }; const fetchNotification = useCallback(async (id: string) => { if (!token || !userId) return; try { const response = await fetch( `${process.env.EXPO_PUBLIC_API_URL}/graphql`, { method: 'POST', headers: { Authorization: `Bearer ${token}`, 'Content-Type': 'application/json', }, body: JSON.stringify({ query: `{ notifications(id: ${id}) { id, alert { id, userId, createdAt, area, extendedArea, level, reachedUsers }, position {id, userId, createdAt, latitude, longitude, movingActivity}, seen, createdAt } }`, }), } ); const data = await response.json(); if (data.errors) { Alert.alert('Error', 'Error fetching data'); } else if (data.data.notifications && data.data.notifications.length > 0) { const notificationData = data.data.notifications[0]; const coordinatesString = notificationData.alert.area.substring(9, notificationData.alert.area.length - 2); const coordinates = coordinatesString .split(',') .map((coord: string) => coord.trim().split(' ')) .map((pair: string[]) => ({ longitude: parseFloat(pair[0]), latitude: parseFloat(pair[1]), })); const extendedCoordinatesString = notificationData.alert.extendedArea.substring(9, notificationData.alert.extendedArea.length - 2); const extendedCoordinates = extendedCoordinatesString .split(',') .map((coord: string) => coord.trim().split(' ')) .map((pair: string[]) => ({ longitude: parseFloat(pair[0]), latitude: parseFloat(pair[1]), })); setCoordinates({ latitude: notificationData.position.latitude, longitude: notificationData.position.longitude }); setNotification(notificationData); setPolygon(coordinates); setExtendedPolygon(extendedCoordinates); setRegion({ latitude: coordinates[0]?.latitude || region.latitude, longitude: coordinates[0]?.longitude || region.longitude, latitudeDelta: 0.05, longitudeDelta: 0.05, }); return notificationData; } else { Alert.alert('Error', 'No data found'); router.push('/notifications/index'); } } catch (err) { console.error('Fetch Map Data Error:', err); } }, [token, userId, region.latitude, region.longitude]); useFocusEffect( useCallback(() => { checkAuth(); }, [checkAuth]) ); const { id } = useLocalSearchParams(); useEffect(() => { if (typeof id === 'string') { fetchNotification(id).then((n) => { if (n && !n.seen) { updateSeenStatus(n.id); } }) } }, [id, fetchNotification]); useEffect(() => { if (mapRef.current && region) { mapRef.current.animateToRegion(region, 1000); } }, [region]); const formatDate = (timestamp: string) => { const date = new Date(parseInt(timestamp, 10) * 1000); return `${date.toDateString()} ${date.getHours()}:${(date.getMinutes() < 10 ? '0' : '') + date.getMinutes()}`; }; const theme = useColorScheme() ?? 'light'; return ( {notification === null ? ( Loading... ) : ( <> Notifications list Notified: {formatDate(notification.createdAt)} Alerted: {formatDate(notification.alert.createdAt)} Level of alert: {notification.alert.level} )} ); } const styles = StyleSheet.create({ map: { height: 400, }, dateRow: { flexDirection: 'row', alignItems: 'center', }, icon: { marginRight: 8, }, });